package com.peng.controller;

import com.peng.entity.Permission;
import com.peng.entity.User;
import com.peng.entity.UserInfo;
import com.peng.utils.JwtUtils;
import com.peng.utils.MenuTree;
import com.peng.utils.Result;
import com.peng.utils.redis.RedisService;
import com.peng.vo.RouterVo;
import com.peng.vo.TokenVo;
import io.jsonwebtoken.Jwts;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.security.core.userdetails.UserDetails;
import org.springframework.security.web.authentication.logout.SecurityContextLogoutHandler;
import org.springframework.util.ObjectUtils;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.List;
import java.util.Objects;
import java.util.function.Predicate;
import java.util.stream.Collectors;

@RestController
@RequestMapping("/api/sysUser")
public class SysUserController {

    @Autowired
    private JwtUtils jwtUtils;

    @Autowired
    private RedisService redisService;

    // 刷新token信息
    @PostMapping("/refreshToken")
    public Result refreshToken(HttpServletRequest request) {
        // 从header中获取token信息
        String token = request.getHeader("token");
        // 判断header是否存在token
        if (ObjectUtils.isEmpty(token)) {
            // 从请求参数中获取token
            token = request.getParameter("token");
        }
        // 从springSecurity上下文中获取用户信息
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
        // 获取用户的身份信息
        UserDetails userDetails = (UserDetails) authentication.getPrincipal();
        // 定义变量，保存新的token信息
        String newToken = "";
        // 验证提交过来的token信息是否合法
        if (jwtUtils.validateToken(token, userDetails)) {
            // 重新生成新的token
            newToken = jwtUtils.refreshToken(token);
        }
        // 获取本次token的到期时间
        long expireTime = Jwts.parser()
                .setSigningKey(jwtUtils.getSecret())
                .parseClaimsJws(newToken.replace("jwt_", ""))
                .getBody().getExpiration().getTime();
        // 清楚原来的token信息
        String oldTokenKey = "token_" + token;
        redisService.del(oldTokenKey);
        // 将新的token信息保存到缓存中
        String newTokenKey = "token_" + newToken;
        redisService.set(newTokenKey, newToken, jwtUtils.getExpiration() / 1000);
        // 创建tokenvo对象
        TokenVo tokenVo = new TokenVo(expireTime, newToken);
        // 返回数据
        return Result.ok(tokenVo).message("token生成成功");
    }

    /**
     * 获取用户信息
     */
    @GetMapping("/getInfo")
    public Result getInfo() {
        // 从Spring Security中获取用户信息
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
        // 判断authentication是否为空
        if (authentication == null) {
            return Result.error().message("用户信息查询失败");
        }
        // 获取用户信息
        User user = (User) authentication.getPrincipal();
        // 获取该用户拥有的权限信息
        List<Permission> permissionList = user.getPermissionList();
        // 获取权限编码
        Object[] roles = permissionList.stream()
                .filter(Objects::nonNull)
                .map(Permission::getCode)
                .toArray();
        // 创建用户信息
        UserInfo userInfo = new UserInfo(user.getId(), user.getNickName(), user.getAvatar(), null, roles);
        // 返回数据
        return Result.ok(userInfo).message("用户信息查询成功");
    }

    /**
     * 获取登录用户的菜单数据
     */
    @GetMapping("/getMenuList")
    public Result getMenuList() {
        // 从Spring Security中获取用户信息
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
        // 获取用户信息
        User user = (User) authentication.getPrincipal();
        // 获取该用户拥有的权限信息
        List<Permission> permissionList = user.getPermissionList();
        //筛选当前用户拥有的目录和菜单数据
        List<Permission> menuList = permissionList.stream()
                // 只筛选目录和菜单数据，按钮不需要添加到路由菜单中
                .filter(item -> item != null && item.getType() != 2)//0-目录 1-菜单 2-按钮
                .collect(Collectors.toList());
        // 生成路由数据
        List<RouterVo> routerVoList = MenuTree.makeRouter(menuList, 0L);
        // 返回数据
        return Result.ok(routerVoList).message("菜单数据获取成功");
    }

    /**
     * 退出登录
     * @param request
     * @param response
     * @return
     */
    @PostMapping("/logout")
    public Result logout(HttpServletRequest request, HttpServletResponse response){
        // 获取token
        String token = request.getHeader("token");
        //如果头部中没有携带token，则从参数中获取
        if (ObjectUtils.isEmpty(token)){
            // 从参数中获取参数信息
            token = request.getParameter("token");
        }
        // 从Spring Security上下文获取用户信息
        Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
        // 判断用户信息是否为，如果不为空，则需要清空用户信息
        if (authentication!=null){
            // 清空用户信息
            new SecurityContextLogoutHandler().logout(request,response,authentication);
            // 清空redis中的token
            redisService.del("token_"+token);
            return Result.ok().message("用户退出成功");
        }
        return Result.error().message("用户退出失败");
    }
}
